diff --git a/landing/competitor-data.js b/landing/competitor-data.js index 7336fbd88..3bd08f059 100644 --- a/landing/competitor-data.js +++ b/landing/competitor-data.js @@ -1,114 +1,114 @@ // @flow const competitors = Object.freeze({ DISCORD: 'discord', KEYBASE: 'keybase', MATRIX: 'matrix', SIGNAL: 'signal', SLACK: 'slack', TELEGRAM: 'telegram', }); export type Competitors = $Values; -type FeatureComparison = { +export type FeatureComparison = { +title: string, +comingSoon: boolean, +competitorDescriptionShort: string, +commDescriptionShort: string, +competitorDescriptionLong: $ReadOnlyArray, +commDescriptionLong: $ReadOnlyArray, +furtherReadingLinks?: $ReadOnlyArray, }; -type Competitor = { +export type Competitor = { +id: Competitors, +name: string, +featureComparison: $ReadOnlyArray, }; const competitorData: $ReadOnlyArray = [ { id: 'signal', name: 'Signal', featureComparison: [ { title: 'Backup', comingSoon: true, competitorDescriptionShort: 'Signal does not back up your data. Data is stored locally on your device.', commDescriptionShort: 'Comm backs up all of your encrypted user data via our backup service.', competitorDescriptionLong: [ 'With the exception of some group membership information, Signal does not back up your data. The only way to transfer data from an old phone to a new phone is via P2P transfer, which is not always possible.', ], commDescriptionLong: [ 'Comm backs up all of your user data via our backup service. The backup is encrypted so that Comm is not able to access the data.', ], furtherReadingLinks: [ 'https://signal.org/blog/signal-private-group-system/', ], }, { title: 'Communities', comingSoon: false, competitorDescriptionShort: 'Signal does not support communities with channels à la Discord or Slack.', commDescriptionShort: 'Comm supports communities with features including channels, roles, threads, and more.', competitorDescriptionLong: [ 'While Signal supports group chats, it does not support communities with channels à la Discord or Slack. There are no user-owned backends on Signal, which limits product functionality and scale.', ], commDescriptionLong: [ 'Comm leverages keyservers to support sophisticated community functionality, including channels, roles, threads, and much more.', ], }, { title: 'Identity', comingSoon: false, competitorDescriptionShort: 'Your identity on Signal is linked to a phone number.', commDescriptionShort: 'Comm accounts are associated with a username or an Ethereum wallet', competitorDescriptionLong: [ 'Your identity on Signal is linked to a phone number, which limits the anonymity and sovereignty of user accounts.', ], commDescriptionLong: [ 'Comm accounts can be linked either to a pseudonymous username or to an Ethereum wallet.', ], }, { title: 'Key resets', comingSoon: false, competitorDescriptionShort: 'Signal’s servers can reset anybody’s public keys in order to facilitate account recovery.', commDescriptionShort: 'Comm backs up user keys, and facilitates account recovery by recovering those original keys.', competitorDescriptionLong: [ 'Signal’s servers have the ability to reset any phone number’s public keys at any time. This functionality is used to facilitate account recovery. Signal relies on users manually verifying “Safety Numbers” in order to verify that a public key reset is valid.', ], commDescriptionLong: [ 'Comm backs up user keys, and facilitates account recovery by recovering those original keys. Note that if the user forgets the secret securing their backup, they will not be able to recover their account.', ], furtherReadingLinks: [ 'https://twitter.com/CommDotApp/status/1545193952554336257', ], }, { title: 'Notifications', comingSoon: false, competitorDescriptionShort: 'Signal has a single function to mute notifs from a chat.', commDescriptionShort: 'Comm allows you to manage notif alerts separately from notif badging.', competitorDescriptionLong: [ 'Signal has a single function to mute notifs from a chat.', ], commDescriptionLong: [ 'Comm allows you to manage notif alerts separately from notif badging (unread icon). Comm also sorts muted chats in a separate “Background” tab in order to avoid cluttering your inbox.', ], }, ], }, ]; export { competitorData }; diff --git a/landing/competitor-feature.css b/landing/competitor-feature.css index f113a2202..e95f3271d 100644 --- a/landing/competitor-feature.css +++ b/landing/competitor-feature.css @@ -1,50 +1,52 @@ .container { display: flex; flex-direction: column; flex: 1; } .headingContainer { display: flex; justify-content: space-between; align-items: center; margin-bottom: 16px; + flex-wrap: wrap; + row-gap: 8px; } .headingText { color: var(--white-100); } .comingSoonBadge { background-color: var(--badge); padding: 5px 10px; border-radius: 16px; } .comingSoonText { color: var(--white-100); margin-left: 8px; } .comingSoonIcon { color: var(--coming-soon-icon); } hr { margin: 24px 0; border-radius: 8px; height: 1.5px; background-color: var(--black-80); border: none; } .descriptionText { color: var(--white-80); - height: 80px; + min-height: 80px; margin-top: 8px; } .descriptionTextMutli { color: var(--white-80); margin-top: 16px; } diff --git a/landing/feature-modal.css b/landing/feature-modal.css new file mode 100644 index 000000000..ecc0c09c0 --- /dev/null +++ b/landing/feature-modal.css @@ -0,0 +1,44 @@ +.modalContainer { + background-color: var(--comparison-cards); + border-radius: 8px; + padding: 24px 32px; + width: 624px; + max-width: 90vw; + max-height: 90vh; + overflow-y: auto; +} + +.featureContainer { + display: flex; +} + +.closeIconContainer { + margin-left: 22px; + cursor: pointer; +} + +.closeIcon { + color: var(--white-80); +} + +.furtherReadingContainer { + border-top: 1px solid var(--black-80); + padding-top: 24px; + margin-top: 24px; +} + +.furtherReadingsText { + color: var(--white-100); + margin-bottom: 16px; +} + +.linksContainer { + display: flex; + flex-direction: column; + row-gap: 8px; +} + +.linkText { + color: var(--violet-dark-100); + text-decoration: underline; +} diff --git a/landing/feature-modal.react.js b/landing/feature-modal.react.js new file mode 100644 index 000000000..fe64b1a5f --- /dev/null +++ b/landing/feature-modal.react.js @@ -0,0 +1,76 @@ +// @flow + +import { faTimes } from '@fortawesome/free-solid-svg-icons'; +import { FontAwesomeIcon } from '@fortawesome/react-fontawesome'; +import classNames from 'classnames'; +import * as React from 'react'; + +import ModalOverlay from 'lib/components/modal-overlay.react.js'; +import { useModalContext } from 'lib/components/modal-provider.react.js'; + +import type { FeatureComparison, Competitor } from './competitor-data.js'; +import CompetitorFeature from './competitor-feature.react.js'; +import css from './feature-modal.css'; +import typography from './typography.css'; + +type Props = { + +competitor: Competitor, + +feature: FeatureComparison, +}; + +function FeatureModal(props: Props): React.Node { + const { competitor, feature } = props; + + const { popModal } = useModalContext(); + + const furtherReadingClassName = classNames( + typography.paragraph1, + css.furtherReadingsText, + ); + const linkClassName = classNames([typography.paragraph2, css.linkText]); + + const furtherReadingLinks = React.useMemo(() => { + if (!feature.furtherReadingLinks) { + return null; + } + + const links = feature.furtherReadingLinks.map((link, index) => ( + + {link} + + )); + + return ( +
+

Further reading

+
{links}
+
+ ); + }, [feature.furtherReadingLinks, furtherReadingClassName, linkClassName]); + + return ( + +
+
+ +
+ +
+
+ {furtherReadingLinks} +
+
+ ); +} + +export default FeatureModal; diff --git a/landing/global.css b/landing/global.css index 1aac812ee..eb75d155f 100644 --- a/landing/global.css +++ b/landing/global.css @@ -1,77 +1,78 @@ :root { --landing-page-z-index: 0; --mobile-nav-z-index: 1; --header-z-index: 2; --purple: #7e57c2; --white: #fff; --white1: #ebedee; --grey: #808080; --btn-bg: var(--purple); --unselected: var(--grey); --btn-color: var(--white1); --logo-color: var(--white); --sub-heading-color: var(--purple); --sans-serif: 'IBM Plex Sans', sans-serif; --white-100: #ffffff; --white-90: #f5f5f5; --white-80: #ebebeb; --white-70: #e0e0e0; --white-60: #cccccc; --black-100: #0a0a0a; --black-90: #1f1f1f; --black-80: #404040; --black-70: #666666; --black-60: #808080; --violet-dark-100: #7e57c2; --violet-dark-80: #6d49ab; --violet-dark-60: #563894; --violet-dark-40: #44297a; --violet-dark-40: #331f5c; --success-light-10: #d5f6e3; --success-light-50: #6cdf9c; --success-primary: #00c853; --success-dark-50: #029841; --success-dark-90: #034920; --error-light-10: #feebe6; --error-light-50: #f9947b; --error-primary: #f53100; --error-dark-50: #b62602; --error-dark-90: #4f1203; --page-background: #111111; --light-dark-page-background: #1a1a1a; --comparison-cards: #18181a; --comparison-cards-hovered: #1c1c1f; --coming-soon-icon: #dca008; --badge: #ebebeb14; + --modal-overlay: rgba(0, 0, 0, 0.8); } /* ===== GENERAL PAGE STYLES ===== */ html, body, :global(div#react-root) { height: 100%; } p { font-size: clamp(0.75rem, 0.5408rem + 1.0458vw, 1.75rem); } a { transition: 0.2s; text-decoration: none; } @media screen and (-webkit-min-device-pixel-ratio: 0) and (min-resolution: 0.001dpcm) { img { image-rendering: -webkit-optimize-contrast !important; } } @media not all and (min-resolution: 0.001dpcm) { @supports (-webkit-appearance: none) and (stroke-color: transparent) { img { image-rendering: unset !important; } } }